Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: Add Fastly Edge SDK #723

Merged
merged 52 commits into from
Mar 10, 2025
Merged

Conversation

ldhenry
Copy link
Contributor

@ldhenry ldhenry commented Jan 3, 2025

Requirements

  • I have added test coverage for new or changed functionality
  • I have followed the repository's pull request submission guidelines
  • I have validated my changes against all supported platform versions

Describe the solution you've provided

This PR adds a new Fastly Compute SDK. This SDK must be used in conjunction with our upcoming Fastly KV integration. The SDK is essentially the same as the other edge integrations (Akamai, Cloudflare, Vercel), with the following changes:

  • node:events is not compatible with Fastly's runtime. As a result, we cannot use @launchdarkly/sdk-server-edge. Instead, I copied the contents of @launchdarkly/sdk-server-edge into the fastly package and replaced createCallbacks.ts with an empty implementation.
  • Enabled sendEvents by default.
  • Added the eventsUri option to allow for sending events to a custom endpoint.
  • Added a new optional eventsBackendName option. A Fastly Backend configured to https://events.launchdarkly.com is required for sending events. The default value is launchdarkly. This option is passed to Fastly's customized fetch().

I added an example app that demonstrates using the SDK to evaluate a feature flag edge to control the static image served.

I published an beta version to npm here.

Copy link
Contributor

github-actions bot commented Jan 3, 2025

@launchdarkly/js-sdk-common size report
This is the brotli compressed size of the ESM build.
Size: 19014 bytes
Size limit: 21000

Copy link
Contributor

github-actions bot commented Jan 3, 2025

@launchdarkly/js-client-sdk size report
This is the brotli compressed size of the ESM build.
Size: 19871 bytes
Size limit: 21000

Copy link
Contributor

github-actions bot commented Jan 3, 2025

@launchdarkly/js-client-sdk-common size report
This is the brotli compressed size of the ESM build.
Size: 15371 bytes
Size limit: 20000

@ldhenry ldhenry marked this pull request as ready for review January 3, 2025 17:58
@ldhenry ldhenry requested a review from a team as a code owner January 3, 2025 17:58
@ldhenry ldhenry requested a review from kinyoklion January 3, 2025 17:58
@@ -0,0 +1,46 @@
import { internal } from '@launchdarkly/js-server-sdk-common';
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

All tests in this directory were copied over from @launchdarkly/sdk-server-edge

@@ -0,0 +1,59 @@
import { PlatformData, SdkData } from '@launchdarkly/js-server-sdk-common';
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This was copied from @launchdarkly/sdk-server-edge

@mmrj
Copy link
Contributor

mmrj commented Jan 3, 2025

(docs ticket is at DOCS-973 and currently scheduled for week of 1/6)

@@ -0,0 +1,116 @@
import type {
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This was copied from @launchdarkly/sdk-server-edge

@@ -0,0 +1,9 @@
const createCallbacks = () => ({
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is the new empty implementation to get around the lack of node:events

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

So I think we can make a task, not blocking this, which removes the dependency from the common package.

That would then allow for the eventually removal of most of the code for this package.

@@ -0,0 +1,17 @@
import { BasicLogger, LDOptions } from '@launchdarkly/js-server-sdk-common';
Copy link
Contributor Author

@ldhenry ldhenry Jan 3, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This mostly copied from @launchdarkly/sdk-server-edge

@@ -0,0 +1,49 @@
import CryptoJS from 'crypto-js';
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Everything in this directory was copied from @launchdarkly/sdk-server-edge

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Do we know if we need crypto-js, or does fastly compute have its own suitable alternatives?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It seems like they have explicitly tested crypto-js as noted here. With that being said, I'll do a test to see if we can get away without it.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I did some digging and it looks like Fastly exposes SimpleCrypto. I started to implement our Hasher with this but hit a roadblock because SimpleCrypto's digest function returns a promise.

I'll stick with crypto-js for now unless you are aware of a workaround.


fetch(url: string, options: Options = {}): Promise<Response> {
// @ts-ignore
return fetch(url, { ...options, backend: this.eventsBackend });
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is the main differentiator between the other edge SDKs. We are required to pass a fastly-specific backend parameter.

Copy link
Member

@kinyoklion kinyoklion left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I've not looked through everything yet, but I did do an initial pass with a few items. Sorry for the delay.

@@ -0,0 +1,9 @@
const createCallbacks = () => ({
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

So I think we can make a task, not blocking this, which removes the dependency from the common package.

That would then allow for the eventually removal of most of the code for this package.

@@ -0,0 +1,49 @@
import CryptoJS from 'crypto-js';
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Do we know if we need crypto-js, or does fastly compute have its own suitable alternatives?

@ldhenry
Copy link
Contributor Author

ldhenry commented Jan 17, 2025

I've not looked through everything yet, but I did do an initial pass with a few items. Sorry for the delay.

No problem and I appreciate your feedback. I'll take another pass next week.

@ldhenry ldhenry requested a review from keelerm84 March 3, 2025 19:25
Copy link
Contributor

@tanderson-ld tanderson-ld left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

General question: why are there a handful of files that are copied from common and not used as dependencies?

return {
name,
version,
userAgentBase: 'FastlyEdgeSDK',
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Do you want to distinguish server sdk from a future client sdk?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think Edge is probably enough of a distinction here. I'm happy to change it but I don't see any future client SDK for Fastly (or other edge providers). I was going off of Cloudflare, and Vercel.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yeah, I think you can think of Edge as an equivalent level to Client or Server. It theoretically could be possible to make a 'Client' SDK for edge, but it wouldn't be very useful.


if (!logger) {
throw new Error('You must configure the client with a logger');
}
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think the logger is optional on most SDKs.

   * If you do not set this property, the SDK uses {@link basicLogger} with a
   * minimum level of `info`.
   */
  logger?: LDLogger;

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We are always setting a logger here if not specified so the additional validation in validateOptions is maybe a little redundant.


if (!featureStore || typeof featureStore !== 'object' || !featureStore.get) {
throw new Error('You must configure the client with a feature store');
}
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Here is the logic used by the type validators for the feature store. https://github.com/launchdarkly/js-core/blob/main/packages/shared/common/src/validators.ts#L22

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks. I started down this road and it seems promising. I have to sign off soon but I should have your suggestion in first thing tomorrow.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I just updated it to use type validators. Thanks for pointing this out.

@kinyoklion
Copy link
Member

kinyoklion commented Mar 4, 2025

General question: why are there a handful of files that are copied from common and not used as dependencies?

@tanderson-ld This is because the common edge package has dependencies which are not suitable (and shouldn't have been included originally).

Basically node:events is used in the common edge package and should not have been. The correct way to fix this is to remove that dependency and migrate it into the edge SDKs that can use it, or major version them and just remove it entirely. It isn't actually useful functionality in the edge.

Added: SDK team should figure this out at some point, but not block progress on this PR.

@ldhenry
Copy link
Contributor Author

ldhenry commented Mar 4, 2025

General question: why are there a handful of files that are copied from common and not used as dependencies?

@tanderson-ld This is because the common edge package has dependencies which are not suitable (and shouldn't have been included originally).

Basically node:events is used in the common edge package and should not have been. The correct way to fix this is to remove that dependency and migrate it into the edge SDKs that can use it, or major version them and just remove it entirely. It isn't actually useful functionality in the edge.

@kinyoklion , @tanderson-ld, IMO a major version bump that removes node:events seems reasonable to me. Unfortunately I don't think I'll have the bandwidth for this before I go on parental leave. Do you think this is a blocker for the Fastly SDK?

@ldhenry ldhenry requested a review from tanderson-ld March 5, 2025 17:54
@@ -14,6 +14,7 @@ export interface EdgeProvider {

export class EdgeFeatureStore implements LDFeatureStore {
private readonly _rootKey: string;
private _kvData: string | null | undefined;
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I believe only undefined is necessary for this situation. Is there another API that has both null and undefined that is causing you to use both here?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The Fastly KV fetch method returns null when the KV item does not exist and we have a null check in the initialized method. However, I don't think undefined is needed so I removed that.

* This function is used to lazy load the KV data from the edge provider. This is necessary because Fastly Compute
* has a limit of 32 backend requests (including requests to fetch the KV data).
* https://docs.fastly.com/products/compute-resource-limits
*/
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Do you want to describe in the readme or public APIs that data won't receive updates during the runtime of the fastly process/execution/request handler lifecycle?

(I think I have that right)

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks for the feedback. I just added a Usage Notes section to the README that contains this.

@ldhenry ldhenry requested a review from tanderson-ld March 7, 2025 20:52
@ldhenry ldhenry dismissed kinyoklion’s stale review March 10, 2025 15:55

Got approval from Todd

@ldhenry ldhenry merged commit 02e0eee into main Mar 10, 2025
24 checks passed
@ldhenry ldhenry deleted the hbarrow/REL-4756/create-fastly-sdk-in-js-core branch March 10, 2025 16:05
@github-actions github-actions bot mentioned this pull request Mar 10, 2025
ldhenry pushed a commit that referenced this pull request Mar 10, 2025
🤖 I have created a release *beep* *boop*
---


<details><summary>fastly-server-sdk: 0.1.0</summary>

##
[0.1.0](fastly-server-sdk-v0.0.1...fastly-server-sdk-v0.1.0)
(2025-03-10)


### Features

* Add Fastly Edge SDK
([#723](#723))
([02e0eee](02e0eee))
</details>

---
This PR was generated with [Release
Please](https://github.com/googleapis/release-please). See
[documentation](https://github.com/googleapis/release-please#release-please).

Co-authored-by: github-actions[bot] <41898282+github-actions[bot]@users.noreply.github.com>
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

5 participants